用PyQt4+Python写一个简单的EPub阅读器(3/3)

最后一篇,这次我们加上几个函数,使得这个阅读器能够按照章节阅读电子书,前两篇我们将完成了Book模块,实现了电子书的数据抽象,然后写了GUI,完成了一个简单的功能,在仓库中插入图书,把LibraryTableWidget刷新,使得图书在library呈现,可以看到书名,作者。接下来我们完善这个程序。

  • 载入书籍的章节
    效果是这样的:

首先,我们在BookView模块中加入这样一个函数:

def load_book(self, book_id):
    self.book = Book(book_id)
    self.chapter_list.clear()
    for chapter in self.book.chapters:
        self.chapter_list.addItem(chapter[0])
    self.chapter_list.setCurrentRow(0)

根据book_id(也就是epub文件名),刷新chapter_list的内容,把每个章节的名称作为QListWidget的item添加进来。

然后,什么时候让这个函数起作用,怎么让这个函数起作用?我们要求的逻辑是双击Library的Item,然后载入这本书的章节。这里就要用到Qt的信号槽机制了,网上关于Qt信号槽机制的文章很多,书本上也有介绍,大多都是balabala说一大堆,堆概念堆名词,我尝试用几句话来说说这个机制的核心。其实它跟IFTTT(if this then that)这个工具是很像的,一个对象发送信号,另一个对象接收到信号就做出相应的动作,为什么要这样设计呢?其实我们可以尝试思考Qt设计者的初衷,因为这个过程是符合人类思考过程的,能够让编程更简单,如果了解了其他的GUI库的事件处理方式,如Java的监听机制,WFT的消息映射,就能发觉Qt的signal/slot机制有多好了,理解了这一点,再去看这方面的资料就能得心应手。

这里没有谈Qt信号槽的细节,我只是提出一种思考方式,因为我也是初学者,但这种思考方式能让我们在遇到稍特别的细节时会不感到诧异,比如,连接信号和槽的connect函数可以使得信号和信号连接,一个button发送了一个click()信号,于是asignal()信号也发送了,就如一个人头疼,他没有直接叫医生,而是通知了朋友,让朋友帮忙叫医生,又比如一个信号可以和多个槽连接,就如一个人头疼,他发了一条朋友圈,于是当医生的朋友都来了。。。。这些过程都是容易理解的,是符合人类思考过程的。

好了,瞎扯淡环节结束,我们继续。让chapter_list的双击Item事件与book_view的load_book函数绑定,我们可以在LibraryTableWidget中加上:

def create_connections(self):
    self.connect(self, SIGNAL("itemDoubleClicked(QTableWidgetItem *)"), self.view_book)

def view_book(self):
    book_id = self.library['books'][self.currentRow()]['id']
    self.book_view.load_book(book_id)

当然,也需要在LibraryTableWidget, __init__函数中加上 :

self.create_connections()    

完成之后,再运行main.py就能实现上面展示的内容。

  • web_view显示图书内容的部分
    其实跟上面的方式大同小异,依葫芦画瓢, 我就直接给出代码了:

在Book模块中加上

def get_chapter(self, num):
    return self.f.read(self.oebps_folder+self.chapters[num][1])

根据章节数目获得html文件内容

在BookView中加上

def set_chapter(self, num=None):
    if num is None:
        num = self.chapter_list.currentRow()
    if num < 0:
        num = len(self.book.chapters) - 1
    elif num >= len(self.book.chapters):
        num = 0
    self.web_view.setHtml(self.book.get_chapter(num).decode(encoding="utf-8"))

使得web_view可以根据章节数目显示电子书中该章节的内容

同样在BookView的__init__函数中加上:

self.create_connections()

绑定signal/slot, 在BookView中加上create_connection函数:

def create_connections(self):
    chlist = self.chapter_list
    self.connect(self.next_button, SIGNAL("clicked()"), lambda:
                 chlist.setCurrentRow(0
                     if chlist.currentRow() == chlist.count() - 1
                     else chlist.currentRow() + 1))
    self.connect(self.previous_button, SIGNAL("clicked()"), lambda:
                 chlist.setCurrentRow(chlist.count() - 1
                                      if chlist.currentRow() == 0
                                      else chlist.currentRow() - 1))
    self.connect(self.chapter_list, SIGNAL("currentRowChanged(int)"),
                 self.set_chapter)

    page = self.web_view.page()
    page.setLinkDelegationPolicy(QWebPage.DelegateAllLinks)
    self.connect(page, SIGNAL("linkClicked(const QUrl&)"),
                 self.link_clicked)

最终的效果:

最后,给出源代码地址